/*-----------------------------------------------------------------------------
Copyright (C) 2011 SIL International
Responsibility: Tim Eves

    This library is free software; you can redistribute it and/or modify
    it under the terms of the GNU Lesser General Public License as published
    by the Free Software Foundation; either version 2.1 of License, or
    (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
    Lesser General Public License for more details.

    You should also have received a copy of the GNU Lesser General Public
    License along with this library in the file named "LICENSE".
    If not, write to the Free Software Foundation, 51 Franklin Street,
    Suite 500, Boston, MA 02110-1335, USA or visit their web page on the
    internet at http://www.fsf.org/licenses/lgpl.html.

Alternatively, the contents of this file may be used under the terms of the
Mozilla Public License (http://mozilla.org/MPL) or the GNU General Public
License, as published by the Free Software Foundation, either version 2
of the License or (at your option) any later version.

Description:
The test harness for the Endian.h template library.  This validates the
Endian.h classes are working correctly and with all fundamental C/C++ integer
types. This may also be used for benchmarking by suppling the base two
magnitude for the number of rounds to run.
Adding -DHAVE_64_LONG to your compilers command line will test 64bit wide
integers in addition to 32, 16 and 8 bit.
-----------------------------------------------------------------------------*/
#include <cstdlib>
#include <string>
#include "inc/Endian.h"

typedef unsigned char 		uint8;
typedef signed char		     int8;
typedef short unsigned int 	uint16;
typedef short signed int 	 int16;
typedef unsigned int 		uint32;
typedef signed int 		     int32;
#if defined(HAVE_64_LONG)
typedef long unsigned int 	uint64;
typedef long signed int 	 int64;
#else
typedef void                uint64;
typedef void                 int64;
#endif

template <typename T8, typename T4, typename T2, typename T1>
bool test_swaps()
{
	bool t = true;

	t &= be::swap<T1>(T1(0xFEU)) == T1(0xFEU);
	t &= le::swap<T1>(T1(0xFEU)) == T1(0xFEU);

	t &= be::swap<T2>(be::swap<T2>(T2(0xFFEEU))) == T2(0xFFEEU);
	t &= le::swap<T2>(le::swap<T2>(T2(0xFFEEU))) == T2(0xFFEEU);
	t &= be::swap<T2>(T2(0xFFEEU)) == le::swap<T2>(T2(0xEEFFU));
#if defined(HAVE_64_LONG)
	t &= be::swap<T4>(be::swap<T4>(T4(0xDEADBEEFU))) == T4(0xDEADBEEFU);
	t &= le::swap<T4>(le::swap<T4>(T4(0xDEADBEEFU))) == T4(0xDEADBEEFU);
	t &= be::swap<T4>(T4(0xDEADBEEFU)) == le::swap<T4>(T4(0xEFBEADDEU));
#endif
    return t;
}


template <typename T8, typename T4, typename T2, typename T1>
int test_reads(const size_t rounds)
{
	static const unsigned char data[] = {0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF,
	                                     0xEF,0xCD,0xAB,0x89,0x67,0x45,0x23,0x01};
	bool t = true;

#if defined(HAVE_64_LONG)
	for (size_t r = rounds; r; --r)
	{
	    const unsigned char *p = data;
		if (be::read<T8>(p) != T8(0x0123456789ABCDEF))  return  8;
		if (le::read<T8>(p) != T8(0x0123456789ABCDEF))  return 80;
	}
#endif
	for (size_t r = rounds; r; --r)
	{
	    const unsigned char *p = data;
		t  = be::read<T4>(p) == T4(0x01234567);
		t &= be::read<T4>(p) == T4(0x89ABCDEF);
		if (!t) return  4;
		t  = le::read<T4>(p) == T4(0x89ABCDEF);
		t &= le::read<T4>(p) == T4(0x01234567);
		if (!t) return 40;
	}
	for (size_t r = rounds; r; --r)
    {
	    const unsigned char *p = data;
		t  = be::read<T2>(p) == T2(0x0123);
		t &= be::read<T2>(p) == T2(0x4567);
		t &= be::read<T2>(p) == T2(0x89AB);
		t &= be::read<T2>(p) == T2(0xCDEF);
		if (!t) return  2;
		t  = le::read<T2>(p) == T2(0xCDEF);
		t &= le::read<T2>(p) == T2(0x89AB);
		t &= le::read<T2>(p) == T2(0x4567);
		t &= le::read<T2>(p) == T2(0x0123);
		if (!t) return 20;
	}
	for (size_t r = rounds; r; --r)
	{
	    const unsigned char *p = data;
		t  = be::read<T1>(p) == T1(0x01);
		t &= be::read<T1>(p) == T1(0x23);
		t &= be::read<T1>(p) == T1(0x45);
		t &= be::read<T1>(p) == T1(0x67);
		t &= be::read<T1>(p) == T1(0x89);
		t &= be::read<T1>(p) == T1(0xAB);
		t &= be::read<T1>(p) == T1(0xCD);
		t &= be::read<T1>(p) == T1(0xEF);
		if (!t) return  1;
		t  = le::read<T1>(p) == T1(0xEF);
		t &= le::read<T1>(p) == T1(0xCD);
		t &= le::read<T1>(p) == T1(0xAB);
		t &= le::read<T1>(p) == T1(0x89);
		t &= le::read<T1>(p) == T1(0x67);
		t &= le::read<T1>(p) == T1(0x45);
		t &= le::read<T1>(p) == T1(0x23);
		t &= le::read<T1>(p) == T1(0x01);
		if (!t) return 10;
	}
	return 0;
}

int main(int argc , char *argv[])
{
	const size_t rounds = 1UL << atoi(argv[1]);
	int r = 0;

    if   (!test_swaps<uint64, uint32, uint16, uint8>()
       || !test_swaps<int64, int32, int16, int8>())
        return 5;

	if (r == 0) r = test_reads<uint64, uint32, uint16, uint8>(rounds);
	if (r == 0) r = test_reads<int64, int32, int16, int8>(rounds);
	return r;
}
